import {Demo} from '../../lib/components/FloatingArrow';

<PageCard>

# FloatingArrow

Renders a customizable `<svg>{:html}` pointing arrow triangle
inside the floating element that gets automatically positioned.

```js
import {FloatingArrow} from '@floating-ui/react';
```

<ShowFor packages={['react-dom']}>

<PackageLimited>@floating-ui/react only</PackageLimited>

</ShowFor>

</PageCard>

<Chrome center>
  <Demo />
</Chrome>

## Usage

- Create an <WordHighlight id="b">arrowRef</WordHighlight> and
  pass it to the [arrow middleware](/docs/arrow)'s
  `element{:.key}` option and the `<FloatingArrow />{:js}`
  component. This lets the arrow be measured for positioning.
- Pass the <WordHighlight id="a">context</WordHighlight> to the
  `<FloatingArrow />{:js}` component. This lets the component
  read the positioning data.

```js /context/1,3#a /arrowRef/#b
import {FloatingArrow, arrow} from '@floating-ui/react';

function App() {
  const arrowRef = useRef(null);
  const {refs, floatingStyles, context} = useFloating({
    middleware: [
      arrow({
        element: arrowRef,
      }),
    ],
  });

  return (
    <>
      <div ref={refs.setReference} />
      <div ref={refs.setFloating} style={floatingStyles}>
        <FloatingArrow ref={arrowRef} context={context} />
      </div>
    </>
  );
}
```

The arrow will, by default, overlap the reference element. The
`height{:.keyword}` of the arrow will offset it by the desired
amount.

```js
import {offset} from '@floating-ui/react';

const ARROW_HEIGHT = 7;
const GAP = 2;

useFloating({
  middleware: [offset(ARROW_HEIGHT + GAP)],
});
```

This does not take into account tip rounding or strokes.

## Props

The arrow accepts all the props of an `<svg>{:js}` element, plus
some additional props:

```ts
interface Props extends React.SVGAttributes<SVGSVGElement> {
  context: FloatingContext;
  width?: number;
  height?: number;
  tipRadius?: number;
  staticOffset?: number | string | null;

  // Inherited SVG props that are intercepted and passed
  // to the <path>s
  d?: string;
  fill?: string;
  stroke?: string;
  strokeWidth?: number;
}
```

### `ref{:.keyword}`

<Required />

```js
<FloatingArrow ref={arrowRef} />
```

### `context{:.keyword}`

<Required />

The `context{:.const}` object returned from `useFloating(){:js}`.

```js
const {context} = useFloating();
<FloatingArrow context={context} />;
```

### `width{:.keyword}`

default: `14{:js}`

The width of the arrow.

```js
<FloatingArrow ref={arrowRef} context={context} width={10} />
```

### `height{:.keyword}`

default: `7{:js}`

The height of the arrow.

```js
<FloatingArrow ref={arrowRef} context={context} height={10} />
```

### `tipRadius{:.keyword}`

default: `0{:js}` (sharp)

The radius (rounding) of the arrow tip.

```js
<FloatingArrow ref={arrowRef} context={context} tipRadius={2} />
```

### `staticOffset{:.keyword}`

default: `undefined{:js}` (use dynamic position)

A static offset override of the arrow from the floating element
edge. Often desirable if the floating element is smaller than the
reference element along the relevant axis and has an edge
alignment (`'start'{:js}`/`'end'{:js}`).

```js
<FloatingArrow
  ref={arrowRef}
  context={context}
  staticOffset={isEdgeAlignedAndSmaller ? '15%' : null}
/>
```

### `d{:.keyword}`

default: `undefined{:js}` (use dynamic path)

A custom path for the arrow. Useful if you want fancy rounding.
The path should be inside a square SVG and placed at the bottom
of it. The path is designed for the `'bottom'{:js}` placement,
which will be rotated for other placements.

```js
<FloatingArrow
  ref={arrowRef}
  context={context}
  width={20}
  height={20}
  d="M0 20C0 20 2.06906 19.9829 5.91817 15.4092C7.49986 13.5236 8.97939 12.3809 10.0002 12.3809C11.0202 12.3809 12.481 13.6451 14.0814 15.5472C17.952 20.1437 20 20 20 20H0Z"
/>
```

### `fill{:.keyword}`

default: `"black"{:js}` (browser default)

The color of the arrow.

```js
<FloatingArrow ref={arrowRef} context={context} fill="red" />
```

### `stroke{:.keyword}`

default: `"none"{:js}`

The stroke (border) color of the arrow. This must match (or be
less than) the floating element's border width.

```js
<FloatingArrow ref={arrowRef} context={context} stroke="red" />
```

### `strokeWidth{:.keyword}`

default: `0{:js}`

The stroke (border) width of the arrow.

```js
<FloatingArrow
  ref={arrowRef}
  context={context}
  strokeWidth={2}
/>
```

## Tailwind and utility CSS styling

- `fill-*{:.string}` sets the arrow's fill color.
- `[&>path:first-of-type]{:.string}` targets the "stroke" path.
- `[&>path:last-of-type]{:.string}` targets the "fill" path's
  extra stroke, to reduce gaps.

`strokeWidth{:.keyword}` should still be manually specified as a
prop.

```js
<FloatingArrow
  ref={arrowRef}
  context={context}
  className="
    fill-white 
    [&>path:first-of-type]:stroke-pink-500
    [&>path:last-of-type]:stroke-white
  "
/>
```

## Scale transforms

When animating the floating element's scale, it looks best if the
floating element's `transform-origin` is at the tip of the arrow.
The `arrow` middleware provides data to achieve this.

[View on CodeSandbox](https://codesandbox.io/s/floating-ui-react-scale-transform-origin-qv0t1c?file=/src/App.tsx)
